int smp_suspend(void)
{
- int i, err;
-
- lock_cpu_hotplug();
-
- /*
- * Take all other CPUs offline. We hold the hotplug mutex to
- * avoid other processes bringing up CPUs under our feet.
- */
- while (num_online_cpus() > 1) {
- unlock_cpu_hotplug();
- for_each_online_cpu(i) {
- if (i == 0)
- continue;
- err = cpu_down(i);
- if (err) {
- printk(KERN_CRIT "Failed to take all CPUs "
- "down: %d.\n", err);
- for_each_possible_cpu(i)
- vcpu_hotplug(i);
- return err;
- }
+ int cpu, err;
+
+ for_each_online_cpu(cpu) {
+ if (cpu == 0)
+ continue;
+ err = cpu_down(cpu);
+ if (err) {
+ printk(KERN_CRIT "Failed to take all CPUs "
+ "down: %d.\n", err);
+ for_each_possible_cpu(cpu)
+ vcpu_hotplug(cpu);
+ return err;
}
- lock_cpu_hotplug();
}
return 0;
{
int cpu;
- for_each_possible_cpu(cpu)
- cpu_initialize_context(cpu);
-
- unlock_cpu_hotplug();
-
for_each_possible_cpu(cpu)
vcpu_hotplug(cpu);
}
pfn_to_mfn(xen_start_info->store_mfn);
xen_start_info->console.domU.mfn =
pfn_to_mfn(xen_start_info->console.domU.mfn);
+ } else {
+ extern cpumask_t cpu_initialized_map;
+ cpu_initialized_map = cpumask_of_cpu(0);
}
set_fixmap(FIX_SHARED_INFO, xen_start_info->shared_info);
}
#endif
- err = smp_suspend();
- if (err)
- return err;
+ for (;;) {
+ err = smp_suspend();
+ if (err)
+ return err;
+
+ xenbus_suspend();
+ preempt_disable();
- xenbus_suspend();
+ if (num_online_cpus() == 1)
+ break;
- preempt_disable();
+ preempt_enable();
+ xenbus_suspend_cancel();
+ }
mm_pin_all();
local_irq_disable();
EXPORT_SYMBOL(cpu_online_map);
cpumask_t cpu_possible_map;
EXPORT_SYMBOL(cpu_possible_map);
-static cpumask_t cpu_initialized_map;
+cpumask_t cpu_initialized_map;
struct cpuinfo_x86 cpu_data[NR_CPUS] __cacheline_aligned;
EXPORT_SYMBOL(cpu_data);
cpu_idle();
}
-void cpu_initialize_context(unsigned int cpu)
+static void cpu_initialize_context(unsigned int cpu)
{
vcpu_guest_context_t ctxt;
struct task_struct *idle = idle_task(cpu);
struct Xgt_desc_struct *gdt_descr = &per_cpu(cpu_gdt_descr, cpu);
#endif
- if (cpu == 0)
+ if (cpu_test_and_set(cpu, cpu_initialized_map))
return;
memset(&ctxt, 0, sizeof(ctxt));
if (rc)
return rc;
- if (!cpu_isset(cpu, cpu_initialized_map)) {
- cpu_set(cpu, cpu_initialized_map);
- cpu_initialize_context(cpu);
- }
+ cpu_initialize_context(cpu);
if (num_online_cpus() == 1)
alternatives_smp_switch(1);
#if defined(CONFIG_HOTPLUG_CPU)
-#if defined(CONFIG_X86)
-void cpu_initialize_context(unsigned int cpu);
-#else
-#define cpu_initialize_context(cpu) ((void)0)
-#endif
-
int cpu_up_check(unsigned int cpu);
void init_xenbus_allowed_cpumask(void);
int smp_suspend(void);